home *** CD-ROM | disk | FTP | other *** search
/ CD ROM Paradise Collection 4 / CD ROM Paradise Collection 4 1995 Nov.iso / program / swags_z.zip / SCREEN.SWG / 0075_Universal Fastwrite.pas < prev    next >
Pascal/Delphi Source File  |  1995-03-03  |  4KB  |  106 lines

  1. (*
  2. Re:    Copy of: Submission for SWAG
  3.  
  4. Here is a unit I have developed for very fast direct screen writes.  
  5. This unit is, unless you know better, the fastest direct direct write 
  6. procedure available.  This is because it is optimised to use word 
  7. size moves on word boundaries where possible. 
  8.  
  9. It also works in both real and protected modes. 
  10. *)
  11.  
  12. {$S-}
  13.  
  14. UNIT FWrite;
  15.  
  16. INTERFACE
  17.  
  18.   PROCEDURE FastWrite(Col, Row, Attr : Byte; Str : String);
  19.   {Fast Direct Screen Writes for Real/Protected Mode Programs.    }
  20.   {Note: 'Col' and 'Row' are zero relative, ie:- 0,0 for Top-Left.}
  21.  
  22. IMPLEMENTATION
  23.  
  24. USES
  25.   Crt;
  26.  
  27.   PROCEDURE FastWrite(Col, Row, Attr : Byte; Str : String); ASSEMBLER;
  28.   ASM
  29.     PUSH   DS           {Save DS}
  30.     MOV    DL,CheckSnow {Save CheckSnow Setting}
  31.     MOV    ES,SegB800   {ES = Colour Screen Segment}
  32.     MOV    SI,SegB000   {SI = Mono Screen Segment}
  33.     MOV    DS,Seg0040   {DS = ROM Bios Segment}
  34.     MOV    BX,[49h]     {BL = CRT Mode, BH = ScreenWidth}
  35.     MOV    AL,Row       {AL = Row No}
  36.     MUL    BH           {AX = Row * ScreenWidth}
  37.     XOR    CH,CH        {CH = 0}
  38.     MOV    CL,Col       {CX = Column No}
  39.     ADD    AX,CX        {(Row*ScreenWidth)+Column}
  40.     ADD    AX,AX        {Multiply by 2 (2 Byte per Position)}
  41.     MOV    DI,AX        {DI = Screen Offset}
  42.     CMP    BL,7         {CRT Mode = Mono?}
  43.     JNE    @@DestSet    {No  - Use Colour Screen Segment}
  44.     MOV    ES,SI        {Yes - ES = Mono Screen Segment}
  45.     XOR    DX,DX        {Force jump to FWrite}
  46.   @@DestSet:            {ES:DI = Screen Destination Address}
  47.     LDS    SI,Str       {DS:SI = Source String}
  48.     CLD                 {Move Forward through String}
  49.     LODSB               {Get Length Byte of String}
  50.     MOV    CL,AL        {CX = Input String Length}
  51.     JCXZ   @@Done       {Exit if Null String}
  52.     MOV    AH,Attr      {AH = Attribute}
  53.     OR     DL,DL        {Test Mono/CheckSnow Flag}
  54.     JZ     @@FWrite     {Snow Checking Disabled or Mono - Use FWrite}
  55. {Output during Screen Retrace's}
  56.     MOV    DX,003DAh    {6845 Status Port}
  57.   @@WaitLoop:           {Output during Retrace's}
  58.     MOV    BL,[SI]      {Load Next Character into BL}
  59.     INC    SI           {Update Source Pointer}
  60.     CLI                 {Interrupts off}
  61.   @@Wait1:              {Wait for End of Retrace}
  62.     IN      AL,DX       {Get 6845 status}
  63.     TEST    AL,8        {Vertical Retrace in Progress?}
  64.     JNZ     @@Write     {Yes - Output Next Char}
  65.     SHR     AL,1        {Horizontal Retrace in Progress?}
  66.     JC      @@Wait1     {Yes - Wait until End of Retrace}
  67.   @@Wait2:              {Wait for Start of Next Retrace}
  68.     IN      AL,DX       {Get 6845 status}
  69.     SHR     AL,1        {Horizontal Retrace in Progress?}
  70.     JNC     @@Wait2     {No - Wait until Retrace Starts}
  71.   @@Write:              {Output Char and Attribute}
  72.     MOV     AL,BL       {Put Char to Write into AL}
  73.     STOSW               {Store Character and Attribute}
  74.     STI                 {Interrupts On}
  75.     LOOP   @@WaitLoop   {Repeat for Each Character}
  76.     JMP    @@Done       {Exit}
  77. {Ignore Screen Retrace's}
  78.   @@FWrite:             {Output Ignoring Retrace's}
  79.     TEST   SI,1         {DS:SI an Even Offset?}
  80.     JZ     @@Words      {Yes - Skip (On Even Boundary)}
  81.     LODSB               {Get 1st Char}
  82.     STOSW               {Write 1st Char and Attrib}
  83.     DEC    CX           {Decrement Count}
  84.     JCXZ   @@Done       {Finished if only 1 Char in Str}
  85.   @@Words:              {DS:SI Now on Word Boundary}
  86.     SHR    CX,1         {CX = Char Pairs, Set CF if Odd Byte Left}
  87.     JZ     @@ChkOdd     {Skip if No Pairs to Store}
  88.   @@Loop:               {Loop Outputing 2 Chars per Loop}
  89.     MOV    BH,AH        {BH = Attrib}
  90.     LODSW               {Load 2 Chars}
  91.     XCHG   AH,BH        {AL = 1st Char, AH = Attrib, BH = 2nd Char}
  92.     STOSW               {Store 1st Char and Attrib}
  93.     MOV    AL,BH        {AL = 2nd Char}
  94.     STOSW               {Store 2nd Char and Attrib}
  95.     LOOP   @@Loop       {Repeat for Each Pair of Chars}
  96.   @@ChkOdd:             {Check for Final Char}
  97.     JNC    @@Done       {Skip if No Odd Char to Display}
  98.     LODSB               {Get Last Char}
  99.     STOSW               {Store Last Char and Attribute}
  100.   @@Done:               {Finished}
  101.     POP    DS           {Restore DS}
  102.   END; {FastWrite}
  103.  
  104. END.
  105.  
  106.